home *** CD-ROM | disk | FTP | other *** search
/ START Magazine / START VOL 3 NO 12.st / BABEL.ARC / BABEL.BAS next >
Encoding:
BASIC Source File  |  1989-04-12  |  12.6 KB  |  458 lines

  1. REM Program Babel
  2. REM by Jim Pierson-Perry 
  3. REM Copright 1989 Antic Publishing
  4. REM 
  5. REM
  6. REM Babel is a sound file conversion program. It accepts sound
  7. REM files created by Replay4, Digisound, Navarone/Hippo ST Sound
  8. REM Designer, Macintosh SoundCap/SoundMaster, Amiga IFF and Mac/ST
  9. REM versions of Sound Designer. Output sound files can be written
  10. REM in Replay4, Digisound or Navarone/Hippo formats. 
  11. REM
  12. REM Program functions are load a sound file, play the sound (audition)
  13. REM through monitor speaker or Replay4/Digisound cartridge, adjust replay
  14. REM rate and save sound file in one of the three supported formats. A demo
  15. REM of sound creation via pure math is provided using the Karplus-Strong
  16. REM string synthesis algorithm.
  17. REM
  18. REM This program was coded using Hisoft BASIC Professional using the
  19. REM Replay4 binary file extension to the Hisoft compiler library (comes
  20. REM with the Replay4 program disk from MichTron). All screen
  21. REM resolutions are supported.
  22. REM
  23. REM *************************************************************************
  24. REM
  25. REM Start with definitions and variable initializations.
  26. REM Force all variable to default to 2-byte integer words
  27. REM
  28. defint a-z
  29. REM
  30. REM Compiler needs to pull routines from the Hisoft library.
  31. REM GEMAES and GEMDOS come with Hisoft. The REPLAY library 
  32. REM came from the Replay4 program disk and was added into the
  33. REM Hisoft compiler library using its BUILDLIB.TTP program.
  34. REM
  35. library "gemaes","gemdos","replay"
  36. dim mess(7),nav_rates(7)
  37. REM
  38. REM Use constants as reference points for menu items in the
  39. REM working subroutines. If the menu is changed later, only these
  40. REM constants need to change (rather than each reference within
  41. REM the program)
  42. REM
  43. const m_about=9,m_load=18,m_save=19,m_shift=21,m_quit=25,m_audition=23,m_form=40
  44. const m_rate5=27,m_audio=36,m_ks=20,mask=255,mu_mesag=16,mn_selected=10
  45. rrate=4
  46. audio_out=0
  47. save_format=0
  48. data 5000,7500,10000,15000,20000,31000,31000,31000
  49. for i=0 to 7
  50. read nav_rates&(i)
  51. next i
  52. sample$=chr$(rrate)
  53. nsamp&=0
  54. valid_sample=0
  55. REM
  56. REM Build the program menu
  57. REM
  58. menu$="[ Desk |  About Babel ]"
  59. menu$=menu$+"[ File |  Load Sample \  Save Sample \  String Synth \  Shift Base \(---------------\  Audition \(---------------\  Quit ]"
  60. menu$=menu$+"[ Rate |   5   \   7.5 \  10   \  15   \  20   \  31   \  40   \  50   ]"
  61. menu$=menu$+"[ Audio |  Monitor Speaker \  Replay Cartridge \(-------------------\  Save Format... \"
  62. menu$=menu$+"     ...Replay4 \     ...Digisound \     ...Navarone ]"
  63. REM
  64. REM Activate the menu, disable desk accessories and set defaults
  65. REM for replay rate and audio output
  66. REM  
  67. m_pointer&=FNmenu&(menu$)
  68. for i=m_about+2 to m_about+7
  69.     menu_ienable m_pointer&,i,0
  70. next i
  71. menu_icheck m_pointer&,m_rate5+rrate,1
  72. menu_icheck m_pointer&,m_audio,1
  73. menu_icheck m_pointer&,m_form,1
  74. menu_ienable m_pointer&,m_rate5+6,0
  75. menu_ienable m_pointer&,m_rate5+7,0
  76. REM
  77. REM Main program loop. It cycles here looking for action on the menu
  78. REM bar. FNevnt_multi used to make it easy to add ST keyboard command
  79. REM equivalents for menu actions in a subsequent version
  80. REM
  81. do
  82.     e=FNevnt_multi(mu_mesag,0,0,0,0,0,0,0,0,0,0,0,0,0,varptr(mess(0)),0,0,0,0,0,k,0)
  83.     if e and mu_mesag then do_menu mess(0)
  84. loop
  85. REM
  86. REM Main subroutine for determining what program action to take as
  87. REM a result of triggering the menu bar
  88. REM
  89. sub do_menu(VAL mes_type)
  90.     shared mess(1),m_pointer&
  91.     static title,item
  92.     if mes_type=mn_selected then
  93.         title=mess(3)
  94.         item=mess(4)
  95.         select case item
  96.             case m_about: call do_about
  97.             case m_load: call do_load
  98.             case m_save: call do_save
  99.             case m_ks: call do_ks
  100.             case m_shift: call do_shift
  101.             case m_audition: call do_audition
  102.             case m_quit: stop -1
  103.             case m_rate5 to m_rate5+7: do_rates item
  104.             case m_audio to m_audio+1: do_audio item
  105.             case m_form to m_form+2: do_format item
  106.         end select
  107.         menu_tnormal m_pointer&,title,1
  108.     end if
  109. end sub
  110. REM
  111. REM Display info box
  112. REM
  113. sub do_about
  114.     static dummy
  115.     dummy=FNform_alert(1,"[0][   Babel |  | by Jim Pierson-Perry | Copyright 1989 Antic Publishing ][ OK ! ]")
  116. end sub
  117. REM
  118. REM Adjust the replay rate. Turn off the check on the old menu
  119. REM item, change the rate and indicate new setting on the menu
  120. REM
  121. sub do_rates(m)
  122.     shared rrate,m_pointer&,audio_out,sample$
  123.     static curr_rate
  124.     if (audio_out=1) or ((m<m_rate5+6) and audio_out=0) then
  125.         menu_icheck m_pointer&,m_rate5+curr_rate+4,0
  126.         curr_rate=m-m_rate5-4
  127.         rrate=m-m_rate5
  128.         menu_icheck m_pointer&,m,1
  129.         mid$(sample$,1,1)=chr$(rrate)
  130.     end if
  131. end sub
  132. REM
  133. REM Set the audio output destination - monitor speaker or through
  134. REM the Replay4/Digisound cartridge to an external speaker
  135. REM
  136. sub do_audio(m)
  137.     shared m_pointer&,audio_out
  138.     static curr_audio
  139.     menu_icheck m_pointer&,curr_audio+m_audio,0
  140.     curr_audio=m-m_audio
  141.     audio_out=curr_audio
  142.     menu_icheck m_pointer&,m,1
  143.     menu_ienable m_pointer&,m_rate5+6,m-m_audio
  144.     menu_ienable m_pointer&,m_rate5+7,m-m_audio
  145. end sub
  146. REM
  147. REM Play the sample data currently in memory using the Replay4
  148. REM library functions extensions to Hisoft BASIC. The valid_sample
  149. REM flag ensures a sample has been loaded so you can't play
  150. REM "empty memory"
  151. REM
  152. sub do_audition
  153.     shared sample$,rrate,audio_out,valid_sample,nsamp&
  154.     if valid_sample=1 then
  155.         hifi audio_out
  156.         frequency rrate
  157.         replay sadd(sample$),nsamp&,0
  158.     end if
  159. end sub
  160. REM
  161. REM Subroutine to set the type of file format to use when saving
  162. REM sample data files to disk. The current choice is updated on
  163. REM the menu bar
  164. REM
  165. sub do_format(m)
  166.     shared m_pointer&,save_format
  167.     static curr_format
  168.     menu_icheck m_pointer&,curr_format+m_form,0
  169.     curr_format=m-m_form
  170.     save_format=curr_format
  171.     menu_icheck m_pointer&,m,1
  172. end sub
  173. REM
  174. REM Subroutine to synthesize a string sample
  175. REM from random numbers (Karplus-Strong algorithm).
  176. REM This is an ST adaptation of the procedure
  177. REM given in Electronic Musician (12/1987)
  178. REM which, in turn, came from an earlier article
  179. REM in Polyphony magazine (12/1984).
  180. REM It is provided as an example of creating
  181. REM a realistic sound from pure math.
  182. REM
  183. sub do_ks
  184.     shared nsamp&,rrate,valid_sample,sample$
  185.     static np,d,l,m,samp_val,i&
  186.     np=64
  187.     d=FNform_alert(0,"[0][   Karplus-Strong |String Synthesis Demo |  |Select Wavetable Size ][ 256 | 128 | 64 ]")
  188.     d=512/(2^d)
  189.     dim rn_table(d)
  190.     mouse 2
  191.     randomize int(timer)
  192.     for l=1 to d
  193.         rn_table(l)=int(rnd(l)*d)
  194.     next l
  195.     nsamp&=clng(np*d)
  196.     sample$=chr$(rrate)+string$(nsamp&,128)
  197.     i&=sadd(sample$)+1&
  198.     for l=1 to np
  199.         for m=2 to d
  200.             samp_val=(rn_table(m)+rn_table(m-1))/2
  201.             rn_table(m-1)=samp_val
  202.             pokeb i&,samp_val
  203.             incr i&
  204.         next m
  205.         samp_val=(rn_table(1)+rn_table(d))/2
  206.         rn_table(d)=samp_val
  207.         pokeb i&,samp_val
  208.         incr i&
  209.     next l
  210.     mouse 0
  211.     erase rn_table
  212.     valid_sample=1
  213. end sub
  214. REM
  215. REM Routine to switch sound file in memory between signed and
  216. REM unsigned integer bases. Needed at minimum by an Amiga sound
  217. REM file type (not IFF) with headerless file structure, like
  218. REM Replay4, but uses signed integer base so it must be adjusted.
  219. REM
  220. sub do_shift
  221.     shared sample$,nsamp&,valid_sample
  222.     static i&,j&,d
  223.     if valid_sample>0 then
  224.         mouse 2
  225.         i&=sadd(sample$)+1&
  226.         j&=i&+nsamp&-1&
  227.         do
  228.             d=(peekb(i&)+128) and mask
  229.             pokeb i&,d
  230.             incr i&
  231.         loop until i&>j&
  232.         mouse 0
  233.     end if
  234. end sub
  235. REM
  236. REM Save the current sample data file from memory to disk using 
  237. REM the currently active save format. The sample in memory is
  238. REM stored in Replay4 format to be compatible with the playback
  239. REM driver used by the audition menu command
  240. REM
  241. sub do_save
  242.     shared sample$,save_format,rrate,nsamp&,nav_rates()
  243.     static fout$,i&,j&,d,ext$
  244.     ext$="SPL"
  245.     if save_format>0 then ext$="SND"
  246.     fout$=FNselect_file$(ext$)
  247.     if fout$="" then exit sub
  248.     open fout$ for output as #1
  249.     mouse 2
  250.     REM
  251.     REM Convert from unsigned to signed integers for Digisound
  252.     REM and Navarone file formats
  253.     REM
  254.     if save_format>0 then
  255.         i&=sadd(sample$)+1&
  256.         j&=i&+nsamp&-1&
  257.         do
  258.             d=(peekb(i&)-128) and mask
  259.             pokeb i&,d
  260.             incr i&
  261.         loop until i&>j&
  262.     end if
  263.     REM
  264.     REM Write the sample with the appropriate header information
  265.     REM
  266.     select case save_format
  267.         case 0
  268.             print #1,mid$(sample$,1&,nsamp&+1&)
  269.         case 1
  270.             print #1,"GP"+mkl$(nsamp&)+chr$(abs(rrate-7))+mid$(sample$,1&,nsamp&+1&)+string$(640,128)+string$(8,0)
  271.         case 2
  272.             print #1,mkl$(nsamp&)+mki$(nav_rates(rrate))+mid$(sample$,2&,nsamp&)
  273.     end select
  274.     close #1
  275.     mouse 0
  276. end sub
  277. REM
  278. REM Subroutine to get a sample data file from disk. It uses internal
  279. REM pattern recognition to determine the incoming file type (e.g. a
  280. REM known header formulation or file length). The default file type
  281. REM is Replay4 as it has no header or anything but the sample data bytes
  282. REM
  283. sub do_load
  284.     shared sample$,valid_sample,nsamp&,rrate
  285.     static i&,j&,k&,flen&,fin$,sbffr$,stype,s_start&,d,m,e
  286.     fin$=FNselect_file$("*")
  287.     if fin$="" then exit sub
  288.     mouse 2
  289.     open fin$ for input as #1
  290.     flen&=lof(1)
  291.     sbffr$=input$(flen&,1)
  292.     close #1
  293.     sample$=chr$(rrate)
  294.     stype=0
  295.     s_start&=1
  296.     nsamp&=flen&
  297.     REM
  298.     REM This section was put in for a particular type of input file
  299.     REM that I use, but is probably not of much interest to others
  300.     REM It recognizes data files from the IBM MIDIUM sample editor
  301.     REM for the Mirage sampler. These have no identifying header but
  302.     REM are always of a set file length.
  303.     REM
  304.     if flen&=66161& then
  305.         d=FNform_alert(1,"[2][ | | Is this a MIDIUM data file ][ Yes | No ]")
  306.         if d=1 then
  307.             stype=6
  308.             s_start&=626
  309.             nsamp&=65536&
  310.         end if
  311.     end if
  312.     REM
  313.     REM Digisound file import
  314.     REM
  315.     if mid$(sbffr$,1,2)="GP" then
  316.         stype=1
  317.         s_start&=9
  318.         nsamp&=cvl(mid$(sbffr$,3,4))
  319.     end if
  320.     REM
  321.     REM Mac SoundCap/SoundMaster file import
  322.     REM
  323.     if mid$(sbffr$,66,7)="FSSDSFX" then
  324.         stype=4
  325.         s_start&=129
  326.         nsamp&=flen&-128&
  327.     end if
  328.     REM
  329.     REM Sound Designer (Mac, without the 128 MacBinary header,
  330.     REM  or ST) file import
  331.     REM
  332.     if cvi(mid$(sbffr$,1,2))=1336 then
  333.         stype=5
  334.         s_start&=1337
  335.         nsamp&=cvl(mid$(sbffr$,185,4))/2&
  336.     end if
  337.     REM
  338.     REM ST Sound Digitizer (Navarone/Hippo)file import
  339.     REM
  340.     if cvl(mid$(sbffr$,1,4))=flen&-6& then
  341.         stype=2
  342.         s_start&=7
  343.         nsamp&=flen&-6&
  344.     end if
  345.     REM
  346.     REM Amiga IFF file import
  347.     REM
  348.     if mid$(sbffr$,1,4)="FORM" then
  349.         stype=3
  350.         s_start&=instr(sbffr$,"BODY")+8&
  351.         nsamp&=cvl(mid$(sbffr$,s_start&-4&,4))
  352.     end if
  353.     select case stype
  354.         case 0
  355.             sample$=sample$+mid$(sbffr$,s_start&,flen&)
  356.         REM
  357.         REM Convert some files from signed integers to unsigned
  358.         REM integers (used by Replay4)
  359.         REM
  360.         case 1 to 3
  361.             sample$=sample$+mid$(sbffr$,s_start&,flen&)
  362.             i&=sadd(sample$)+1&
  363.             j&=i&+nsamp&-1&
  364.             do
  365.                 d=(peekb(i&)+128) and mask
  366.                 pokeb i&,d
  367.                 incr i&
  368.             loop until i&>j&
  369.         case 4
  370.             sample$=sample$+string$(flen&,128)
  371.             i&=sadd(sbffr$)+s_start&-1&
  372.             j&=sadd(sample$)+1&
  373.             d=1
  374.             REM Mac and Amiga.IFF need to have the sample rate
  375.             REM taken down by a factor of 10/11 to compensate
  376.             REM for basic differences in sampling rates supported
  377.             REM by the machines
  378.             REM
  379.             for k&=s_start& to flen&
  380.                 pokeb j&,peekb(i&)
  381.                 incr i&
  382.                 incr j&
  383.                 incr d
  384.                 if d=6 then
  385.                     incr i&
  386.                     d=1
  387.                     nsamp&=nsamp&-1&
  388.                 end if
  389.             next k&    
  390.         case 5
  391.             sample$=sample$+string$(nsamp&,128)
  392.             i&=sadd(sbffr$)+1336&
  393.             j&=sadd(sample$)+1&
  394.             k&=i&+flen&-1
  395.             REM
  396.             REM Sound Designer files must be converted from signed
  397.             REM to unsigned integers. In addition, they have to be
  398.             REM taken from 16 bit resolution down to 8 bit resolution
  399.             REM
  400.             do
  401.                 d=(peekb(i&)+128) and mask
  402.                 pokeb j&,d
  403.                 incr i&
  404.                 incr i&
  405.                 incr j&
  406.             loop until i&>k&
  407.         case 6
  408.             sample$=sample$+mid$(sbffr$,s_start&,flen&)
  409.             i&=sadd(sample$)+1&
  410.             j&=i&+nsamp&
  411.             REM
  412.             REM The MIDIUM files are particularly warped. Each byte
  413.             REM must have the high and low nybbles reversed
  414.             REM
  415.             while i&<j&
  416.                 e=peekb(i&)
  417.                 d=e and 15
  418.                 m=e and 240
  419.                 pokeb i&,16%*d+m/16%
  420.                 incr i&
  421.             wend
  422.     end select
  423.     valid_sample=1
  424.     mouse 0
  425. end sub
  426. REM
  427. REM General routine to obtain file/path name from the GEM file
  428. REM selector box. This was taken almost verbatim from the Hisoft
  429. REM BASIC example disk. I added the ext$ parameter and detection
  430. REM of Cancel button by a null file name
  431. REM
  432. DEF FNselect_file$(ext$)
  433.     static path$,name$,drv$,where,but
  434.     path$=space$(64)
  435.     drv$=chr$(FNdgetdrv+"A"%)
  436.     dgetpath sadd(path$),0
  437.     if left$(path$,1)=chr$(0) then
  438.         path$=drv$+":\*."+ext$
  439.     else
  440.         path$=drv$+":"+path$
  441.         where=instr(path$,chr$(0))
  442.         path$=left$(path$,where-1)
  443.         path$=path$+"\*."+ext$
  444.     end if
  445.     fsel_input path$,name$,but
  446.     cls
  447.     if but=0 then
  448.         FNselect_file$=""
  449.         exit def
  450.     end if
  451.     where=instr(path$,"*")
  452.     path$=left$(path$,where-1)
  453.     FNselect_file$=path$+name$
  454. end def
  455.  
  456.  
  457.  
  458.